package com.longtop.framework.sys.file.supports;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;

import org.apache.commons.io.FileUtils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import com.longtop.framework.util.StringUtil;

public class FileUtil {

	private static final Log log = LogFactory.getLog(FileUtil.class);

	/**
	 * 私有构造方法，防止类的实例化，因为工具类不需要实例化。
	 */
	private FileUtil() {

	}

	/**
	 * 修改文件的最后访问时间。 如果文件不存在则创建该文件。 <b>目前这个方法的行为方式还不稳定，主要是方法有些信息输出，这些信息输出是否保留还在考虑中。</b>
	 * 
	 * @param file
	 *            需要修改最后访问时间的文件。
	 */
	public static void touch(File file) {
		long currentTime = System.currentTimeMillis();
		if (!file.exists()) {
			log.error("file not found:" + file.getName());
			log.debug("Create a new file:" + file.getName());
			try {
				if (file.createNewFile()) {
					log.info("Succeeded!");
				} else {
					log.debug("Create file failed!");
				}
			} catch (IOException e) {
				e.printStackTrace();
			}
		}
		boolean result = file.setLastModified(currentTime);
		if (!result) {
			log.error("touch failed: " + file.getName());
		}
	}

	/**
	 * 修改文件的最后访问时间。 如果文件不存在则创建该文件。 <b>目前这个方法的行为方式还不稳定，主要是方法有些信息输出，这些信息输出是否保留还在考虑中。</b>
	 * 
	 * @param fileName
	 *            需要修改最后访问时间的文件的文件名。
	 */
	public static void touch(String fileName) {
		File file = new File(fileName);
		touch(file);
	}

	/**
	 * 修改文件的最后访问时间。 如果文件不存在则创建该文件。 <b>目前这个方法的行为方式还不稳定，主要是方法有些信息输出，这些信息输出是否保留还在考虑中。</b>
	 * 
	 * @param files
	 *            需要修改最后访问时间的文件数组。
	 */
	public static void touch(File[] files) {
		for (int i = 0; i < files.length; i++) {
			touch(files[i]);
		}
	}

	/**
	 * 修改文件的最后访问时间。 如果文件不存在则创建该文件。 <b>目前这个方法的行为方式还不稳定，主要是方法有些信息输出，这些信息输出是否保留还在考虑中。</b>
	 * 
	 * @param fileNames
	 *            需要修改最后访问时间的文件名数组。
	 */
	public static void touch(String[] fileNames) {
		File[] files = new File[fileNames.length];
		for (int i = 0; i < fileNames.length; i++) {
			files[i] = new File(fileNames[i]);
		}
		touch(files);
	}

	/**
	 * 判断指定的文件是否存在。
	 * 
	 * @param fileName
	 *            要判断的文件的文件名
	 * @return 存在时返回true，否则返回false。
	 */
	public static boolean isFileExist(String fileName) {
		return new File(fileName).isFile();
	}

	/**
	 * 创建指定的目录。 如果指定的目录的父目录不存在则创建其目录书上所有需要的父目录。 <b>注意：可能会在返回false的时候创建部分父目录。</b>
	 * 
	 * @param file
	 *            要创建的目录
	 * @return 完全创建成功时返回true，否则返回false。
	 */
	public static boolean makeDirectory(File file) {
		File parent = file.getParentFile();
		if (parent != null) {
			return parent.mkdirs();
		}
		return false;
	}

	/**
	 * 创建指定的目录。 如果指定的目录的父目录不存在则创建其目录书上所有需要的父目录。 <b>注意：可能会在返回false的时候创建部分父目录。</b>
	 * 
	 * @param fileName
	 *            要创建的目录的目录名
	 * @return 完全创建成功时返回true，否则返回false。
	 */
	public static boolean makeDirectory(String fileName) {
		File file = new File(fileName);
		return makeDirectory(file);
	}

	/**
	 * 清空指定目录中的文件。 这个方法将尽可能删除所有的文件，但是只要有一个文件没有被删除都会返回false。 另外这个方法不会迭代删除，即不会删除子目录及其内容。
	 * 
	 * @param directory
	 *            要清空的目录
	 * @return 目录下的所有文件都被成功删除时返回true，否则返回false.
	 */
	public static boolean emptyDirectory(File directory) {
		boolean result = false;
		File[] entries = directory.listFiles();
		for (int i = 0; i < entries.length; i++) {
			if (!entries[i].delete()) {
				result = false;
			}
		}
		return true;
	}

	/**
	 * 清空指定目录中的文件。 这个方法将尽可能删除所有的文件，但是只要有一个文件没有被删除都会返回false。 另外这个方法不会迭代删除，即不会删除子目录及其内容。
	 * 
	 * @param directoryName
	 *            要清空的目录的目录名
	 * @return 目录下的所有文件都被成功删除时返回true，否则返回false。
	 */
	public static boolean emptyDirectory(String directoryName) {
		File dir = new File(directoryName);
		return emptyDirectory(dir);
	}

	/**
	 * 删除指定目录及其中的所有内容。
	 * 
	 * @param dirName
	 *            要删除的目录的目录名
	 * @return 删除成功时返回true，否则返回false。
	 */
	public static boolean deleteDirectory(String dirName) {
		return deleteDirectory(new File(dirName));
	}

	/**
	 * 删除指定目录及其中的所有内容。
	 * 
	 * @param dir
	 *            要删除的目录
	 * @return 删除成功时返回true，否则返回false。
	 */
	public static boolean deleteDirectory(File dir) {
		if ((dir == null) || !dir.isDirectory()) {
			throw new IllegalArgumentException("Argument " + dir + " is not a directory. ");
		}
		File[] entries = dir.listFiles();
		int sz = entries.length;
		for (int i = 0; i < sz; i++) {
			if (entries[i].isDirectory()) {
				if (!deleteDirectory(entries[i])) {
					return false;
				}
			} else {
				if (!entries[i].delete()) {
					return false;
				}
			}
		}
		if (!dir.delete()) {
			return false;
		}
		return true;
	}

	/**
	 * 列出目录中的所有内容，包括其子目录中的内容。
	 * 
	 * @param fileName
	 *            要列出的目录的目录名
	 * @return 目录内容的文件数组。
	 */
	public static File[] listAll(String fileName) {
		return listAll(new File(fileName));
	}

	/**
	 * 列出目录中的所有内容，包括其子目录中的内容。
	 * 
	 * @param file
	 *            要列出的目录
	 * @return 目录内容的文件数组。
	 */
	@SuppressWarnings("unchecked")
	public static File[] listAll(File file) {
		ArrayList list = new ArrayList();
		File[] files;
		if (!file.exists() || file.isFile()) {
			return null;
		}
		list(list, file, null);
		list.remove(file);
		files = new File[list.size()];
		list.toArray(files);
		return files;
	}

	/**
	 * 列出目录中的所有内容，包括其子目录中的内容。
	 * 
	 * @param file
	 *            要列出的目录
	 * @param filter
	 *            过滤器
	 * @return 目录内容的文件数组。
	 */
	@SuppressWarnings("unchecked")
	public static File[] listAll(File file, javax.swing.filechooser.FileFilter filter) {
		ArrayList list = new ArrayList();
		File[] files;
		if (!file.exists() || file.isFile()) {
			return null;
		}
		list(list, file, filter);
		files = new File[list.size()];
		list.toArray(files);
		return files;
	}

	/**
	 * 将目录中的内容添加到列表。
	 * 
	 * @param list
	 *            文件列表
	 * @param filter
	 *            过滤器
	 * @param file
	 *            目录
	 */
	@SuppressWarnings("unchecked")
	private static void list(ArrayList list, File file, javax.swing.filechooser.FileFilter filter) {
		if (filter != null) {
			if (filter.accept(file)) {
				list.add(file);
				if (file.isFile()) {
					return;
				}
			}
		} else {
			list.add(file);
			if (file.isFile()) {
				return;
			}
		}
		if (file.isDirectory()) {
			File files[] = file.listFiles();
			for (int i = 0; i < files.length; i++) {
				list(list, files[i], filter);
			}
		}

	}

	/**
	 * 返回文件的URL地址。
	 * 
	 * @param file
	 *            文件
	 * @return 文件对应的的URL地址
	 * @throws MalformedURLException
	 * @deprecated 在实现的时候没有注意到File类本身带一个toURL方法将文件路径转换为URL。 请使用File.toURL方法。
	 */
	public static URL getURL(File file) throws MalformedURLException {
		String fileURL = "file:/" + file.getAbsolutePath();
		URL url = new URL(fileURL);
		return url;
	}

	/**
	 * 从文件路径得到文件名。
	 * 
	 * @param filePath
	 *            文件的路径，可以是相对路径也可以是绝对路径
	 * @return 对应的文件名
	 */
	public static String getFileName(String filePath) {
		File file = new File(filePath);
		return file.getName();
	}

	/**
	 * 从文件名得到文件绝对路径。
	 * 
	 * @param fileName
	 *            文件名
	 * @return 对应的文件路径
	 */
	public static String getFilePath(String fileName) {
		File file = new File(fileName);
		return file.getAbsolutePath();
	}

	/**
	 * 将DOS/Windows格式的路径转换为UNIX/Linux格式的路径。 其实就是将路径中的"\"全部换为"/"，因为在某些情况下我们转换为这种方式比较方便， 某中程度上说"/"比"\"更适合作为路径分隔符，而且DOS/Windows也将它当作路径分隔符。
	 * 
	 * @param filePath
	 *            转换前的路径
	 * @return 转换后的路径
	 */
	public static String toUNIXpath(String filePath) {
		return filePath.replace('\\', '/');
	}

	/**
	 * 从文件名得到UNIX风格的文件绝对路径。
	 * 
	 * @param fileName
	 *            文件名
	 * @return 对应的UNIX风格的文件路径
	 * @see #toUNIXpath(String filePath) toUNIXpath
	 */
	public static String getUNIXfilePath(String fileName) {
		File file = new File(fileName);
		return toUNIXpath(file.getAbsolutePath());
	}

	/**
	 * 得到文件的类型。 实际上就是得到文件名中最后一个“.”后面的部分。
	 * 
	 * @param fileName
	 *            文件名
	 * @return 文件名中的类型部分
	 */
	public static String getTypePart(String fileName) {
		int point = fileName.lastIndexOf('.');
		int length = fileName.length();
		if (point == -1 || point == length - 1) {
			return "";
		} else {
			return fileName.substring(point + 1, length);
		}
	}

	/**
	 * 得到文件的类型。 实际上就是得到文件名中最后一个“.”后面的部分。
	 * 
	 * @param file
	 *            文件
	 * @return 文件名中的类型部分
	 */
	public static String getFileType(File file) {
		return getTypePart(file.getName());
	}

	/**
	 * 得到文件的名字部分。 实际上就是路径中的最后一个路径分隔符后的部分。
	 * 
	 * @param fileName
	 *            文件名
	 * @return 文件名中的名字部分
	 */
	public static String getNamePart(String fileName) {
		int point = getPathLsatIndex(fileName);
		int length = fileName.length();
		if (point == -1) {
			return fileName;
		} else if (point == length - 1) {
			int secondPoint = getPathLsatIndex(fileName, point - 1);
			if (secondPoint == -1) {
				if (length == 1) {
					return fileName;
				} else {
					return fileName.substring(0, point);
				}
			} else {
				return fileName.substring(secondPoint + 1, point);
			}
		} else {
			return fileName.substring(point + 1);
		}
	}

	/**
	 * 得到文件名中的父路径部分。 对两种路径分隔符都有效。 不存在时返回""。 如果文件名是以路径分隔符结尾的则不考虑该分隔符，例如"/path/"返回""。
	 * 
	 * @param fileName
	 *            文件名
	 * @return 父路径，不存在或者已经是父目录时返回""
	 */
	public static String getPathPart(String fileName) {
		int point = getPathLsatIndex(fileName);
		int length = fileName.length();
		if (point == -1) {
			return "";
		} else if (point == length - 1) {
			int secondPoint = getPathLsatIndex(fileName, point - 1);
			if (secondPoint == -1) {
				return "";
			} else {
				return fileName.substring(0, secondPoint);
			}
		} else {
			return fileName.substring(0, point);
		}
	}

	/**
	 * 得到路径分隔符在文件路径中首次出现的位置。 对于DOS或者UNIX风格的分隔符都可以。
	 * 
	 * @param fileName
	 *            文件路径
	 * @return 路径分隔符在路径中首次出现的位置，没有出现时返回-1。
	 */
	public static int getPathIndex(String fileName) {
		int point = fileName.indexOf('/');
		if (point == -1) {
			point = fileName.indexOf('\\');
		}
		return point;
	}

	/**
	 * 得到路径分隔符在文件路径中指定位置后首次出现的位置。 对于DOS或者UNIX风格的分隔符都可以。
	 * 
	 * @param fileName
	 *            文件路径
	 * @param fromIndex
	 *            开始查找的位置
	 * @return 路径分隔符在路径中指定位置后首次出现的位置，没有出现时返回-1。
	 */
	public static int getPathIndex(String fileName, int fromIndex) {
		int point = fileName.indexOf('/', fromIndex);
		if (point == -1) {
			point = fileName.indexOf('\\', fromIndex);
		}
		return point;
	}

	/**
	 * 得到路径分隔符在文件路径中最后出现的位置。 对于DOS或者UNIX风格的分隔符都可以。
	 * 
	 * @param fileName
	 *            文件路径
	 * @return 路径分隔符在路径中最后出现的位置，没有出现时返回-1。
	 */
	public static int getPathLsatIndex(String fileName) {
		int point = fileName.lastIndexOf('/');
		if (point == -1) {
			point = fileName.lastIndexOf('\\');
		}
		return point;
	}

	/**
	 * 得到路径分隔符在文件路径中指定位置前最后出现的位置。 对于DOS或者UNIX风格的分隔符都可以。
	 * 
	 * @param fileName
	 *            文件路径
	 * @param fromIndex
	 *            开始查找的位置
	 * @return 路径分隔符在路径中指定位置前最后出现的位置，没有出现时返回-1。
	 */
	public static int getPathLsatIndex(String fileName, int fromIndex) {
		int point = fileName.lastIndexOf('/', fromIndex);
		if (point == -1) {
			point = fileName.lastIndexOf('\\', fromIndex);
		}
		return point;
	}

	/**
	 * 将文件名中的类型部分去掉。
	 * 
	 * @param filename
	 *            文件名
	 * @return 去掉类型部分的结果
	 */
	public static String trimType(String filename) {
		int index = filename.lastIndexOf(".");
		if (index != -1) {
			return filename.substring(0, index);
		} else {
			return filename;
		}
	}

	/**
	 * 得到相对路径。 文件名不是目录名的子节点时返回文件名。
	 * 
	 * @param pathName
	 *            目录名
	 * @param fileName
	 *            文件名
	 * @return 得到文件名相对于目录名的相对路径，目录下不存在该文件时返回文件名
	 */
	public static String getSubpath(String pathName, String fileName) {
		int index = fileName.indexOf(pathName);
		if (index != -1) {
			return fileName.substring(index + pathName.length() + 1);
		} else {
			return fileName;
		}
	}

	/**
	 * 复制文件
	 * 
	 * @param oldFilePath
	 * @param newFilePath
	 * @throws IOException
	 */
	public static void copyFile(String oldFilePath, String newFilePath) throws IOException // 使用FileInputStream和FileOutStream
	{
		FileInputStream fi = new FileInputStream(oldFilePath);
		FileOutputStream fo = new FileOutputStream(newFilePath);
		byte data[] = new byte[fi.available()];
		fi.read(data);
		fo.write(data);
		fi.close();
		fo.close();
	}

	/**
	 * @param filename
	 * @return
	 */
	public static String getFileExtName(String filename) {
		return filename.substring(filename.lastIndexOf(".") + 1, filename.length()).toUpperCase();
	}

	/**
	 * 读取文档正文
	 * 
	 * @param file_path
	 * @return
	 */
	public static String getFileContent(String file_path) {
		try {
			String tempStr = "", tempStr2 = "";
			BufferedReader in = new BufferedReader(new InputStreamReader(new FileInputStream(file_path), "GBK"));
			while ((tempStr = in.readLine()) != null) {
				tempStr2 += tempStr + "\n";
			}
			in.close();
			return tempStr2;
		} catch (Exception e) {
			return "";
		}
	}

	/**
	 * 文件的创建 如果路经不存在，则自动创建路经
	 * 
	 * @return
	 */
	public static boolean createFile(File myFile) throws IOException {
		if (myFile == null) {
			throw new IOException("请指定要创建的文件");
		} else if (myFile.isDirectory()) {
			createDirectory(myFile);
			return true;
		} else {
			try {
				createDirectory(myFile);
				myFile.createNewFile();
				return true;
			} catch (IOException e) {
				throw e;
			}
		}
	}

	/**
	 * 创建文件夹
	 * 
	 * @return
	 */
	public static boolean createDirectory(File myFile) {

		File tempFile = null;
		String tempFilePath = "";
		if (myFile.isFile())
			tempFile = myFile;
		else
			tempFile = myFile.getParentFile();
		String path = StringUtil.getString(tempFile.getAbsolutePath(), "");
		path = StringUtil.replace(path, "\\", "/");
		path = StringUtil.replace(path, "\\", "/");
		String[] fileName = StringUtil.split(path, "/");
		for (int kk = 0; kk < fileName.length; kk++) {
			tempFilePath += fileName[kk] + "/";
			if (!tempFilePath.startsWith("/") && tempFilePath.indexOf(":") != 1) {
				tempFilePath = "/" + tempFilePath;
			}
			// System.out.println("tempFilePath:" + tempFilePath);
			tempFile = new File(tempFilePath);
			tempFile.mkdir();
		}
		return true;
	}

	private static boolean deleteDir(File dir) {
		try {
			FileUtils.deleteDirectory(dir);
			return true;
		} catch (IOException e) {
			e.printStackTrace();
			return false;
		}

	}

	public static void clearDir(File dir) throws IOException {
		FileUtils.cleanDirectory(dir);
	}

	public static void copyDirectiory(String dirSourse, String dirTarget) throws IOException {
		new File(dirSourse).mkdirs();
		File file[] = (new File(dirSourse)).listFiles();
		for (int i = 0; i < file.length; i++) {
			if (file[i].isFile()) {
				FileInputStream input = new FileInputStream(file[i]);
				FileOutputStream output = new FileOutputStream((new StringBuilder(String.valueOf(dirTarget))).append(File.separator).append(file[i].getName()).toString());
				byte b[] = new byte[5120];
				for (int len = 0; (len = input.read(b)) != -1;)
					output.write(b, 0, len);

				output.flush();
				output.close();
				input.close();
			}
			if (file[i].isDirectory())
				copyDirectiory((new StringBuilder(String.valueOf(dirSourse))).append(File.separator).append(file[i].getName()).toString(), (new StringBuilder(String.valueOf(dirTarget))).append(File.separator).append(file[i].getName()).toString());
		}

	}

	public static boolean deleteFile(String path, String fileName) {
		String _file = (new StringBuilder(String.valueOf(path))).append(File.separator).append(fileName).toString();
		File file = new File(_file);
		return deleteDir(file);
	}

	public static boolean writeFile(String dirTarget, String content, String encoding) {
		if (dirTarget == null || dirTarget.length() < 1 || !(new File(dirTarget).isFile())) {
			return false;
		}
		try {
			FileUtils.writeStringToFile(new File(dirTarget), content, encoding);
			return true;
		} catch (IOException e) {
			e.printStackTrace();
			return false;
		}
	}

	public static boolean writeFile(String dirTarget, String content) {
		if (dirTarget == null || dirTarget.length() < 1 || !(new File(dirTarget).isFile())) {
			return false;
		}
		try {
			FileUtils.writeStringToFile(new File(dirTarget), content);
			return true;
		} catch (IOException e) {
			e.printStackTrace();
			return false;
		}
	}

	public static boolean writeFile(File file, byte[] data) {
		if (file == null || !file.isFile()) {
			return false;
		}
		try {
			FileUtils.writeByteArrayToFile(file, data);
			return true;
		} catch (IOException e) {

			e.printStackTrace();
			return false;
		}
	}

	public static boolean writeFile(String dirTarget, byte[] data) {
		if (dirTarget == null || dirTarget.length() < 1 || !(new File(dirTarget).isFile())) {
			return false;
		}
		try {
			FileUtils.writeByteArrayToFile(new File(dirTarget), data);
			return true;
		} catch (IOException e) {
			e.printStackTrace();
			return false;
		}

	}

	public static final URL findAsResource(String path) {
		URL url = null;
		url = Thread.currentThread().getContextClassLoader().getResource(path);
		if (url != null)
			return url;
		url = FileUtil.class.getClassLoader().getResource(path);
		if (url != null)
			return url;
		url = ClassLoader.getSystemClassLoader().getResource(path);
		if (url != null)
			return url;
		if (url == null) {
			File file = new File(path);
			try {
				if (file.exists())
					url = file.toURL();
			} catch (MalformedURLException e) {
				url = null;
			}
		}
		return url;
	}

	public static boolean copy(File srcFile, File destFile, boolean preserveFileDate) throws IOException {
		try {
			FileUtils.copyFile(srcFile, destFile, preserveFileDate);
			return true;
		} catch (IOException e) {
			e.printStackTrace();
			throw new IOException();
		}
	}

	public static boolean copy(File srcFile, File destFile) throws IOException {
		return copy(srcFile, destFile, true);
	}
}